Import CSV

Import CSV 2.3 (27/02/2007)

Ci-joint une version modifiée de l'importation en format CSV des comptes.
Elle prend en compte la nouvelle ligne Détail du format CSV des relevés de la Banque Postale
N'hésitez pas à compléter la liste des idées d'évolution...

-- Fabrice


Fichier concerné
C:\Program Files\BankPerfect\Scripts\Import CSV\import_csv.py

Pour utiliser la modification ci dessous il suffit de copier-coller le code dans le fichier import_csv.py (faites une sauvegarde de l'ancien fichier avant!)

 #_______________________________________________________________________________
 # 2.3 (2007-02-27)
 # Traite correctement les fichiers de la Banqe Postale
 #_______________________________________________________________________________
 import BP
 Acc = BP.AccountCurrent()
 dateFmts, Seps, mapping, Show = ["JMA", "MJA", "AMJ"], [";", ",", "\t"], [0, 0, 0, 0, 0, 0, 0, 0, 0], [50, 100, 500]
 fields = ["Date", "Mode", "Tiers", "Détails", "Catégorie", "Sous-catégorie", "Montant", "Débit", "Crédit"]
 MrkList = ["Non Pointé", "Pointé", "Rapproché"]
 l, b, CC, cs, cb = "TLabel", "TButton", CreateComponent, "csDropDownList", "TComboBox"
 CSV = []
 def Import(S):
   global DateFmt
   for i in range(len(mapping)): mapping[i] = Combos[i].ItemIndex
   if mapping[0] <= 0: s = "La date n'est pas spécifiée !"
   elif (mapping[6] > 0 and mapping[7] > 0) or (mapping[6] > 0 and mapping[8] > 0): s = "Débit/Crédit OU Montant !"
   elif mapping[6] <= 0 and (mapping[7] <= 0 or mapping[8] <= 0): s = "Le montant n'est pas spécifié !"
   else: s = ""
   if s:
     LMsg.Caption = s
     return
   s = EStart.Text
   if s.isdigit(): sLine = int(s)
   else: sLine = 2
   if sLine < 1: sLine = 1
   mark = CBMark.ItemIndex
   for i in range(len(MrkList)):
    if mark == MrkList[i]: mark = i
   s = EEnd.Text
   if s.isdigit(): eLine = int(s)
   else: eLine = 0
   if eLine < sLine: eLine = 0
   DateFmt = dateFmts[CBDate.ItemIndex]
   records, min_date = linestodict(CSV, Seps[CBSep.ItemIndex], sLine, eLine)
   currLines, Light = CurrentLines(min_date)
   idx = CBDup.ItemIndex
   Count = len(records)
   i = 0
   for r in records:
     line = FmtLine(r)
     i += 1
     date, mode, tier, info, catg, mont = r.get("Date", ""), r.get("Mode", ""), r.get("Tiers", ""), r.get("Détails", ""), r.get("Catégorie", ""), r.get("Montant", "")
     if not IsDate(date): report["Dates incorrectes"].append(line)
     elif mont == "" or abs(mont) < 0.0001: report["Montants incorrects"].append(line)
     else:
       insertDupl, idx = TestDup(i, Count, idx, currLines, Light, (date, mode, tier, info, mont), date, simplecheck(mode), mont, line)
       if insertDupl == "ko": continue
       if catg == "": catg = -1
       mode, tier, info = SCheckNum(mode, tier, info)
       count1 = BP.OperationCount[Acc]
       date = Date2Str(date)
       BP.LineAdd(Acc, date, mode, tier, info, catg, mont, mark)
       count2 = BP.OperationCount[Acc]
       if count2 == count1: report["Lignes refusées"].append(line)
       else: report["Opérations importées"].append(line)
   BP.AccountRefreshScreen()
   MRes.Lines.Text = Errors(report)
   FF.ShowModal()
   F0.Close()
#_____________________________________________________________
 def draw(S, ACol, ARow, R, State):
   cv = S.Canvas
   if "gdSelected" in State: cv.Brush.Color = 0x00dec5b9
   elif ARow > 0 and ARow % 2 == 0: cv.Brush.Color = 0x00f5f5f5
   cv.FillRect(R)
   try:
     if ARow == 0:
       if ACol == 0: s = "n°"
       else: s = "Col. %d" %ACol
       cv.Font.Style = ["fsBold"]
     else: s = cells[ARow - 1][ACol]
   except:
     s = ""
   if ACol == 0 and ARow > 0: cv.Font.Color = 0x000000CC
   else: cv.Font.Color = 0x00000000
   if s != "": cv.TextRect(R, R.Left + 3, R.Top + 5, s)
   cv.Brush.Style = 1
#_____________________________________________________________
 def Resize(S):
   Grid.DefaultColWidth = (Grid.Width - 40) / Grid.ColCount
#_____________________________________________________________
 def FillGrid(S):
   global cells
   i = CBShow.ItemIndex
   if i < 3: L = CSV[:Show[i]]
   else: L = CSV
   s = Seps[CBSep.ItemIndex]
   cells = [["%d" %(i + 1)] + l.split(s) for i, l in enumerate(L)]
   cols = max([len(l) for l in cells])
   Grid.ColCount, Grid.RowCount = cols, len(cells)
   for cb in Combos:
     cb.Items.Text = "--\n" + "\n".join([str(i) for i in range(1, cols)])
     cb.ItemIndex = mapping[i]
   Resize(S)
#_____________________________________________________________
 F0 = CC("TForm", None)
 F0.SetProps(Position = "poMainFormCenter", Width=800, Height=480, Caption = "Paramètres de l'import")
 CC(l, F0).SetProps(Parent=F0, Left=20, Top=25, Caption="Date / Sépar. :")
 CC(l, F0).SetProps(Parent=F0, Left=20, Top=50, Caption="Doublons :")
 CC(l, F0).SetProps(Parent=F0, Left=20, Top=75, Caption="Première/dern. ligne :")
 CC(l, F0).SetProps(Parent=F0, Left=20, Top=126, Caption="Pointage :")
 LFld = CC(l, F0)
 LFld.SetProps(Parent=F0, Left=20, Top=155, Caption="Champ")
 LMap = CC(l, F0)
 LMap.SetProps(Parent=F0, Left=130, Top=155, Width=45, Caption="Colonne")
 LMsg = CC(l, F0)
 LMsg.SetProps(Parent=F0, Left=20, Top=410)
 LMsg.Font.Color = 0x000000CC
 CBDate = CC(cb, F0)
 CBDate.SetProps(Parent=F0, Left=130, Top=20, Width=48, Style=cs)
 CBDate.Items.Text = "\n".join(dateFmts)
 CBDate.ItemIndex = 0
 CBSep = CC(cb, F0)
 CBSep.SetProps(Parent=F0, Left=183, Top=20, Width=48, Style=cs, OnChange=FillGrid)
 CBSep.Items.Text = ";\n,\nTAB"
 CBSep.ItemIndex = 0
 CBDup = CC(cb, F0)
 CBDup.SetProps(Parent=F0, Left=130, Top=45, Width=100, Style=cs)
 CBDup.Items.Text = "Tout accepter\nTout refuser\nDemander"
 CBDup.ItemIndex = 2
 EStart = CC("TEdit", F0)
 EStart.SetProps(Parent=F0, Left=130, Top=70, Width=48, Text="2")
 EEnd = CC("TEdit", F0)
 EEnd.SetProps(Parent=F0, Left=183, Top=70, Width=48)
 CCtgs = CC("TCheckBox", F0)
 CCtgs.SetProps(Parent=F0, Left=20, Top=100, Width=200, Caption="Créer les catégories manquantes", Checked=0)
 CBMark = CC(cb, F0)
 CBMark.SetProps(Parent=F0, Left=130, Top=122, Width=100, Style=cs)
 CBMark.Items.Text = "\n".join(MrkList)
 CBMark.ItemIndex = 0
 Grid = CC("TDrawGrid", F0)
 Grid.SetProps(Parent=F0, Left=250, Top=20, Width=525, Height=375, Anchors=["AkLeft", "akTop", "akRight", "akBottom"], FixedCols=0, OnDrawCell=draw, Options = ["goFixedHorzLine", "goVertLine", "goDrawFocusSelected", "goRowSelect", "goThumbTracking"])
 CBShow = CC(cb, F0)
 CBShow.SetProps(Parent=F0, Left=250, Top=407, Width=125, Style=cs, Anchors=["akLeft", "akBottom"], OnChange=FillGrid)
 CBShow.Items.Text = "Afficher 50 lignes\r\nAfficher 100 lignes\r\nAfficher 500 lignes\r\nTout afficher"
 CBShow.ItemIndex = 0
 CC(b, F0).SetProps(Parent=F0, Left=625, Top=405, Width=70, Height=25, Caption="Annuler", Cancel=1, Anchors=["akRight", "akBottom"], ModalResult=2)
 CC(b, F0).SetProps(Parent=F0, Left=705, Top=405, Width=70, Height=25, Caption="Importer", Default=1, OnClick=Import, Anchors=["akRight", "akBottom"])
 F0.OnResize=Resize
 FDup = CC("TForm", None)
 FDup.SetProps(Width=400, Height=390, Position="poMainFormCenter", BorderStyle="bsSingle", BorderIcons=["biSystemMenu"], Caption="Détection de doublons")
 l0 = CC(l, FDup)
 l0.SetProps(Parent=FDup, Left=20, Top=20)
 l1 = CC(l, FDup)
 l1.SetProps(Parent=FDup, Left=20, Top=60, Caption="Ligne à insérer :")
 l2 = CC(l, FDup)
 l2.SetProps(Parent=FDup, Left=20, Top=190, Caption="Ligne existante :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=80, Caption="Date :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=100, Caption="Mode :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=120, Caption="Tiers :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=140, Caption="Détails :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=160, Caption="Montant :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=210, Caption="Date :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=230, Caption="Mode :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=250, Caption="Tiers :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=270, Caption="Détails :")
 CC(l, FDup).SetProps(Parent=FDup, Left=40, Top=290, Caption="Montant :")
 lD = CC(l, FDup)
 lD.SetProps(Parent=FDup, Left=120, Top=80)
 lM = CC(l, FDup)
 lM.SetProps(Parent=FDup, Left=120, Top=100)
 lT = CC(l, FDup)
 lT.SetProps(Parent=FDup, Left=120, Top=120)
 lI = CC(l, FDup)
 lI.SetProps(Parent=FDup, Left=120, Top=140)
 lA = CC(l, FDup)
 lA.SetProps(Parent=FDup, Left=120, Top=160)
 LD = CC(l, FDup)
 LD.SetProps(Parent=FDup, Left=120, Top=210)
 LM = CC(l, FDup)
 LM.SetProps(Parent=FDup, Left=120, Top=230)
 LT = CC(l, FDup)
 LT.SetProps(Parent=FDup, Left=120, Top=250)
 LI = CC(l, FDup)
 LI.SetProps(Parent=FDup, Left=120, Top=270)
 LA = CC(l, FDup)
 LA.SetProps(Parent=FDup, Left=120, Top=290)
 lD.Font.Color = lM.Font.Color = lT.Font.Color = lI.Font.Color = lA.Font.Color = LD.Font.Color = LM.Font.Color = LT.Font.Color = LI.Font.Color = LA.Font.Color = 0x00CC0000
 CC(b, FDup).SetProps(Parent=FDup, Left=40, Top=320, Width=75, Height=25, Caption="Oui", ModalResult=6)
 CC(b, FDup).SetProps(Parent=FDup, Left=120, Top=320, Width=75, Height=25, Caption="Non", ModalResult=7)
 CC(b, FDup).SetProps(Parent=FDup, Left=200, Top=320, Width=75, Height=25, Caption="Toutes", ModalResult=10)
 CC(b, FDup).SetProps(Parent=FDup, Left=280, Top=320, Width=75, Height=25, Caption="Aucune", ModalResult=9)
#_____________________________________________________________
 FF = CC("TForm", None)
 FF.SetProps(Width=640, Height=450, Position="poMainFormCenter", BorderStyle="bsSingle", BorderIcons=["biSystemMenu"], Caption="Résultat de l'import")
 CC(b, FF).SetProps(Parent=FF, Left=270, Top=370, Width=100, Height=25, Caption="Fermer", ModalResult=1, Cancel=1, Default=1)
 MRes = CC("TMemo", FF)
 MRes.SetProps(Parent=FF, Left=30, Top=30, Width=580, Height=350, Anchors=["akTop","akLeft", "akRight", "akBottom"], WordWrap=0, Readonly=1)
 l0.Font.Style = l1.Font.Style = l2.Font.Style = LMap.Font.Style = LMsg.Font.Style = LFld.Font.Style = ["fsBold"]
 F0.Font.Name = FF.Font.Name = FDup.Font.Name = "Tahoma"
#_____________________________________________________________
 def Date2Str(d):
   y, m, d = d
   if y < 100: y += 2000
   return "%02d/%02d/%04d" %(d, m, y)
#_____________________________________________________________
 def Errors(report):
   ok = 0
   ko = 0
   lines = []
   for key in report.keys():
    l = report[key]
    num = len(l)
    if num == 0: continue
    if key == "Opérations importées": ok += num
    else: ko += num
    l = ["  %s" %s for s in l]
    lines.append("%s (%d) :\r\n\r\n%s" %(key, num, "\r\n".join(l)))
   total = ok + ko
   if total == 0:
    s = "Il n'y a aucune ligne à importer depuis le fichier"
   else:
    if ko == 0:
      if total == 1: s = "L'opération contenue dans le fichier CSV a été importée sans erreurs"
      else: s = "Les %d opérations contenues dans le fichier CSV ont été importées sans erreurs" %total
    elif ok == 0: s = "Aucune ligne n'a pu être importée"
    else:
      if ok == 1: s = "Une opération sur %d a été importée" %(total)
      else: s = "%d opérations sur %d ont été importées" %(ok, total)
    s = "%s\r\n\r\n%s" %(s, "\r\n\r\n".join(lines))
   return s
#_____________________________________________________________
 def IsDate(t):
   try:
     y, m, d = t
     if (m == 2) and (y % 4 == 0) and ( (y % 100 != 0) or (y % 400 == 0) ): Max = 29
     else: Max = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31][m - 1]
     return m > 0 and m < 13 and d > 0 and d <= Max
   except:
     return 0
#_____________________________________________________________
 def FmtLine(dic):
   s = []
   for key in dic.keys():
    v = dic[key]
    if key == "Date": v = Date2Str(v)
    s.append("%s : %s" %(key, v))
    s.sort()
   return ",  ".join(s)
#_____________________________________________________________
 def Str2Date(d):
   Seps = "/-' .;"
   d = d.replace("-", "/").replace(".", "/").replace("'", "/").replace(" ", "").split("/")
   if len(d) != 3 or not d[0].isdigit() or not d[1].isdigit() or not d[2].isdigit(): return None
   d = [int(d[0]), int(d[1]), int(d[2])]
   if DateFmt == "JMA": return (d[2], d[1], d[0])
   elif DateFmt == "MJA": return (d[2], d[0], d[1])
   elif DateFmt == "AMJ": return (d[0], d[1], d[2])
   elif DateFmt == "AJM": return (d[0], d[2], d[1])
   else: return None
#_____________________________________________________________
 def Str2Float(value):
   s = value
   s = s.replace("$", "").replace(" ", "").replace("€", "").replace("F", "").replace(",", ".")
   if "." in s:
    l = s.split(".")
    s = "%s.%s" %( "".join(l[:-1]), l[-1] )
   try:
    f = float(s)
   except:
    return None
   return f
#_____________________________________________________________
 def GetCtg(cat, sub):
   cat, sub = cat.strip(), sub.strip()
   if cat == "" and sub == "": return -1
   cat, sub = Pattern(cat), Pattern(sub)
   if cat == "" and sub == "": return -1
   if cat == "" or sub == "":
    t = cat + sub
    if t in CtgPatt: return CtgPatt.index(t)
    else:
      for i in range(len(CtgPatt)):
        c = CtgPatt[i]
        if t in c or c in t > -1: return i
   else:
    match1 = [c for c in categs1 if cat in c[0] or c[0] in cat]
    match2 = [c for c in categs2 if sub in c[0] or c[0] in sub]
    if len(match1) == 0 and len(match2) == 0: t = cat
    else:
      if len(match1) == 0: return match2[0][1]
      if len(match2) == 0: return match1[0][1]
      parent_indexes = [m[1] for m in match1]
      for m in match2:
        if m[2] in parent_indexes: return m[1]
      return match2[0][1]
   if CCtgs.Checked:
     BP.CategAdd(t, 0)
     return BP.CategCount() - 1
   else: return -1
#_____________________________________________________________
 def Pattern(v):
   ko = "ÀÁÂÃÄÅÆÇÈÉÊËÌÍÎÏÑÒÓÔÕÖÙÚÛÜÝßàáâãäåæçèéêëìíîïñòóôõöùúûüýÿABCDEFGHIJKLMNOPQRSTUVWXYZ"
   ok = "aaaaaaaceeeeiiiinooooouuuuybaaaaaaaceeeeiiiinooooouuuuyyabcdefghijklmnopqrstuvwxyz"
   exclude =  " -.,;:/!?'_"
   s = ""
   for c in v:
     i = ko.find(c)
     if i > -1: s += ok[i]
     elif not c in exclude: s += c
   return s
#_____________________________________________________________
 def getmode(value, sign):
   value = value.strip()
   if value != "":
    if sign == -1 and value.isdigit(): return "Chq %s" %value
    v = Pattern(value)
    if sign == 1: fmodes = {"Versement": ["vers", "esp"], "Dépôt de chèque": ["rem", "dep", "ch"], "Virement reçu": ["vir", "vrm", "vrt"]}
    else: fmodes = {"Carte": ["fac", "cb", "car", "achatcarte", "achatcb"], "Retrait DAB": ["ret", "dab", "distrib", "automat"], "Chèque émis": ["ch"], "Prélèvement": ["pr"], "Virement émis": ["vir", "vrm", "vrt"], "TIP": ["tip"]}
    for m in fmodes.keys():
      fpatterns = fmodes[m]
      for fpattern in fpatterns:
        if v.find(fpattern) == 0:
          if m == "Chèque émis":
            v = value.split(" ")[-1].split(".")[-1].split("°")[-1].split("Q")[-1].split("E")[-1].split("H")[-1]
            if v.isdigit(): return "Chq %s" %v
          return m
   if sign == 1: return "Virement reçu"
   else: return "Prélèvement"
#_____________________________________________________________
 def linetodict(line, sep):
   d = {}
   print line
   cols = line.split(sep)
   for i in range(len(fields)):
    field = fields[i]
    num_col = mapping[i] - 1
    if num_col == -1: continue
    if num_col >= len(cols): value = ""
    else: value = cols[num_col]
    if len(value) > 2 and value[0] == '"' and value[-1] == '"': value = value[1:-1]
    if field == "Date": value = Str2Date(value)
    elif (field == "Montant" or field == "Débit" or field == "Crédit") and value != "":
      value = Str2Float(value)
      if field == "Débit": value = -abs(value)
      elif field == "Crédit": value = abs(value)
      if value != None:
        if value > 0: sign = 1
        elif value < 0: sign = -1
        else: sign = 0
      field = "Montant"
    if value != None and field != "Débit" and field != "Crédit": d[field] = value
   d["Catégorie"] = GetCtg(d.get("Catégorie", ""), d.get("Sous-catégorie", ""))
   try:
    if sign != 0: d["Mode"] = getmode(d.get("Mode", ""), sign)
    return d
   except:
    return None
#_____________________________________________________________
 def linestodict(CSV, sep, sLine, eLine):
   global report
   report = {"Opérations importées" : [], "Dates incorrectes" : [], "Montants incorrects" : [], "Enregistrements incorrects" : [], "Lignes refusées" : [], "Doublons" : []}
   min_date, records = (3000, 12, 31), []
   if eLine > 0 and eLine >= sLine: CSV = CSV[sLine - 1:eLine]
   else: CSV = CSV[sLine - 1:]
   for l in CSV:
    if l.strip() == "": continue
    try: record = linetodict(l, sep)
    except: record = None
    if record != None:
      if not record.has_key("Date"): report["Dates incorrectes"].append(l)
      elif not record.has_key("Montant"): report["Montants incorrects"].append(l)
      else:
        y, m, d = record["Date"]
        if y < 100: y += 2000
        record["Date"] = (y, m, d)
        if (y, m, d) < min_date: min_date = (y, m, d)
        records.append(record)
    else: report["Enregistrements incorrects"].append(l)
   return records, min_date
#_____________________________________________________________
 def TestDup(N, T, ask, Lines, Light, R, d, m, v, line):
   if ask == 0: res = "ok", ask
   elif not (d, m, v) in Light: res = "ok", ask
   elif ask == 1: res = "ko", ask
   else:
     i = Light.index((d, m, v))
     l = Lines[i]
     lD.Caption, lM.Caption, lT.Caption, lI.Caption, lA.Caption = Date2Str(R[0]), R[1], R[2], R[3], R[4]
     LD.Caption, LM.Caption, LT.Caption, LI.Caption, LA.Caption = Date2Str(l[0]), l[1], l[3], l[4], l[2]
     l0.Caption = "La ligne %d/%d ressemble à une ligne existante.\nSouhaitez-vous l'insérer ?" %(N, T)
     i = FDup.ShowModal()
     res = {6: ("ok", ask), 10: ("ok", 0), 7: ("ko", ask), 9: ("ko", 1)}[i]
   if res[0] == "ko": report["Doublons"].append(line)
   return res
#_____________________________________________________________
 def simplecheck(s):
   return [s, "Chèque émis"][s.find("Chq") == 0]
#_____________________________________________________________
 def CurrentLines(from_date):
   i = Acc
   lines = zip(BP.OperationDate[i], BP.OperationMode[i], BP.OperationAmount[i], BP.Operationthirdparty[i], BP.OperationDetails[i])
   lines = [((int(dt[6:10]), int(dt[3:5]), int(dt[0:2])), mode, value, people, details) for (dt, mode, value, people, details) in lines if (int(dt[6:10]), int(dt[3:5]), int(dt[0:2])) >= from_date]
   lines_search = [(l[0], simplecheck(l[1]), l[2]) for l in lines]
   return lines, lines_search
#_____________________________________________________________
 def SCheckNum(mode, tier, info):
   if mode == "Chèque émis" and tier != "":
     num = tier.split("°")[-1].split(" ")[-1]
     if num.isdigit(): mode = "Chq %s" %num
     else:
       num = info.split("°")[-1].split(" ")[-1]
       if num.isdigit(): mode = "Chq %s" %num
   if mode.find("Chq ") == 0:
     if info == "": info = "n°%s" %(mode[4:])
     else: info = "n°%s %s" %(mode[4:], info)
     mode = "Chèque émis"
   return mode, tier, info
#_____________________________________________________________
 def guess(mapping, title_line, sep):
   titles = title_line.split(sep)
   titles = [Pattern(t) for t in titles]
   cols = range(len(titles))
   fs = [["date"], ["mode", "type"], ["tiers"], ["details", "note", "notes"], ["categorie", "categ"], ["souscategorie", "sscategorie", "sscateg", "souscateg", "scateg", "scategorie"], ["montant"], ["debit"], ["credit"]]
   for i in range(len(fs)):
     possibles = fs[i]
     for col in cols:
       if titles[col] in possibles:
         mapping[i] = col + 1
         break
#_____________________________________________________________
 CtgPatt = [Pattern(c) for c in BP.CategName]
 parents = BP.CategParent
 categs1 = [(CtgPatt[i], i) for i in range(len(CtgPatt)) if parents[i] == i]
 categs2 = [(CtgPatt[i], i, parents[i]) for i in range(len(CtgPatt)) if parents[i] != i]
 Combos = []
 Path = BP.OpenDialog("Choisissez le fichier à importer", "\\", ".csv", "Comma Separated Values (*.CSV)|*.csv|Vidéoposte (*.TSV)|*.tsv")
 if Path != "":
   if Path[-3:] == "tsv": mapping, EStart.Text, CBSep.ItemIndex = [1, 2, 2, 3, 0, 0, 4, 0, 0], "9", 2
   #<patch v2.1> 02/07/2004
   #CSV = [l.strip() for l in open(Path, "r").readlines()]
   CSVTemp = []
   CSVTemp = [l.strip() for l in open(Path, "r").readlines()]
   sTmpLine = ""
   sSep = ""
   sSep = Seps[CBSep.ItemIndex]
   bAddLine = 0
   iDetailStart = 0
   iDetailEnd = 0
   for l in CSVTemp:
     iDetailStart = l.find(sSep + '"')
     iDetailEnd = l.find('"' + sSep)
     print "iDetailStart : " + str(iDetailStart)
     print "iDetailEnd : " + str(iDetailEnd)
     # Details sur 1 ligne : 
     # 24/01/2007;"CHEQUE N° 234587";-38,00;-249,26
     if (iDetailStart > -1) and (iDetailEnd > -1) and iDetailStart < iDetailEnd:
       sTmpLine = l.replace('"' + sSep, '"' + sSep + sSep)
       l = sTmpLine.replace('"', '')
       sTmpLine = ""
       bAddLine = 0
     # 1ere ligne de Détail
     # 25/01/2007;"ACHAT CB BP NANTERRE    23.01.07
     # CARTE NUMERO                789";-30,00;-196,79
     if iDetailStart > iDetailEnd:
       sTmpLine = l.replace('"', '')
       bAddLine = 1
       continue
     # 2eme ligne de Détail
     if iDetailEnd and bAddLine:
       l = sTmpLine + sSep + l.replace('"', '')
       sTmpLine = ""
       bAddLine = 0
     CSV.append(l)
   #</patch 2.1>
   FillGrid(None)
   cols = "--\n" + "\n".join([str(i) for i in range(1, Grid.ColCount)])
   if sum(mapping) == 0: guess(mapping, CSV[0], Seps[CBSep.ItemIndex])
   for i in range(len(mapping)):
     CC("TLabel", F0).SetProps(Parent=F0, Left=20, Top=i * 25 + 178, Caption=fields[i])
     cb = CC("TComboBox", F0)
     cb.SetProps(Parent=F0, Left=130, Width=50, Top=i * 25 + 175, Style=cs)
     cb.Items.Text = cols
     cb.ItemIndex = mapping[i]
     Combos.append(cb)
   F0.ShowModal()

Historique

2.3 (27/02/07 - Yanns)
Mise à jour des paramètres d'import par défaut des .TSV
2.2 (26/02/07 - Fabrice)
Utilise le séparateur sélectionné et non ";"
2.1 (24/02/07 - Fabrice)
Traite les champs Détails sur 2 lignes encadrés par "
2.0 (Fabio)
Version BankPerfect 2.0.189

Idées d'évolutions / Améliorations

27/02/07

24/02/07

04/03/2007


-- Fabrice